<template>
  <!--判断scopedSlots.default可知道是否存在子元素-->
  <el-table-column
    v-if="$scopedSlots.default"
    :key="$attrs.label"
    v-bind="$attrs"
    :class-name="className"
    :min-width="minWidth"
  >
    <template slot-scope="scope">
      <!--原element组件使用了作用域插槽, 这里仿照-->
      <slot v-bind="scope"></slot>
    </template>
  </el-table-column>
  <!--使用 slot 自定义 header-->
  <el-table-column
    v-else-if="$scopedSlots.header"
    :key="$attrs.label"
    v-bind="$attrs"
    :class-name="className"
    :min-width="minWidth"
  >
    <template slot="header" slot-scope="scope">
      <slot name="header" v-bind="scope"></slot>
    </template>
  </el-table-column>
  <!--默认情况使用原始 el-table-column-->
  <el-table-column
    v-else
    :key="$attrs.label"
    v-bind="$attrs"
    :class-name="className"
    :min-width="minWidth"
  />
</template>

<script>
const $consts = {
  // 字体大小默认14
  fontSize: 14,
  // 计算字体比率
  fontRate: {
    CHAR_RATE: 1.1, // 汉字比率
    NUM_RATE: 0.65, // 数字
    OTHER_RATE: 0.5, // 除汉字和数字以外的字符的比率
  },
};
export default {
  name: "af-table-column",
  data() {
    return {
      minLength: 5, // 初始也不要写成0, 避免未及时设置宽度太丑
      getComputedWidth: 0, // 自定义列中获取元素计算的宽度
    };
  },
  computed: {
    // table数据
    values() {
      const data = this.$parent.data || [];
      return data.map((item) => item[this.$attrs.prop]);
    },
    // 是否自适应列宽, 默认是
    isFit() {
      return (
        (this.$parent.$attrs.autoFit === undefined && this.$attrs.fit === undefined) ||
        (this.$parent.$attrs.autoFit === false && this.$attrs.fit !== undefined)
      );
    },
    // 为存在scope的添加className
    className() {
      const parentClass = this.$attrs["class-name"] || "";
      const classPre = this.$attrs.prop || `encode-${this.transChar(this.$attrs.label)}`; // 有的列因为有slotScope而不给prop
      return classPre ? `${parentClass} ${classPre}-column` : "";
    },
    // 列最小宽度
    minWidth() {
      if (!this.$attrs.label) return undefined;
      if (!this.isFit) return undefined;
      const maxOne =
        Math.max(this.minLength, this.$attrs.label.length * this.fontRate.CHAR_RATE) *
          this.fontSize +
        20;
      return this.$attrs.width || Math.max(maxOne, this.getComputedWidth);
    },
    // 字体大小
    fontSize() {
      return this.$attrs.fontSize || (this.$ELEMENT || {}).fontSize || $consts.fontSize;
    },
    // 字体比例
    fontRate() {
      return {
        ...$consts.fontRate, // 默认值
        ...((this.$ELEMENT || {}).fontRate || {}), // 根注册参数 (Vue.use 时的第二个参数)
        ...(this.$attrs.fontRate || {}), // 父组件属性
      };
    },
  },
  watch: {
    values: {
      immediate: true,
      handler(val) {
        this.isFit !== false &&
          this.$nextTick(() => {
            // 详情中的列表需要在nextTick才能生效
            if (this.$scopedSlots.default) {
              // 存在自定义节点
              setTimeout(() => {
                // 首次获取不到子节点, setTimeout才行
                // 可能存在贴边列, 需要使用包含 fixed 的类名
                const bodyWrapper = this.$attrs.fixed
                  ? document
                      .querySelector(
                        `.el-table__fixed${
                          this.$attrs.fixed === "right" ? `-${this.$attrs.fixed}` : ""
                        }`
                      )
                      .querySelector(".el-table__fixed-body-wrapper")
                  : document.querySelector(".el-table__body-wrapper");
                const nodes = bodyWrapper.querySelectorAll(
                  `.${this.$attrs.prop || `encode-${this.transChar(this.$attrs.label)}`}-column`
                );
                if (nodes.length) {
                  // 有可能会来不及获取nodes, 就切换菜单进入下一页了, 再研究吧
                  const target = [];
                  const getComputedWidths = [];
                  Array.from(nodes).map((item) => {
                    target.push(item.innerText);
                    // 有可能存在自定义列内容超出容器, 取 scrollWidth 才行
                    getComputedWidths.push(item.querySelector(".cell").scrollWidth);
                  });
                  this.getComputedWidth = Math.max(...getComputedWidths);
                  this.$set(this, "minLength", this.getMaxLength(target));
                }
              }, 0);
            } else {
              this.$set(this, "minLength", this.getMaxLength(val));
            }
          });
      },
    },
  },
  methods: {
    // 获取数组中元素按字体比例最长长度
    getMaxLength(arr) {
      return arr.reduce((length, item) => {
        if (item) {
          const str = item.toString();
          const char = str.match(/[\u2E80-\u9FFF]/g);
          const charLength = char ? char.length : 0;
          const num = str.match(/\d|\./g);
          const numLength = num ? num.length : 0;
          const otherLength = str.length - charLength - numLength;
          let newLength =
            charLength * this.fontRate.CHAR_RATE +
            numLength * this.fontRate.NUM_RATE +
            otherLength * this.fontRate.OTHER_RATE;
          if (str.includes("\n")) newLength = this.getMaxLength(str.split("\n"));
          if (length < newLength) {
            length = newLength;
          }
        }
        return length;
      }, 0);
    },
    // 转换汉字为class支持的字母
    transChar(char) {
      return encodeURIComponent(char).replace(/[^a-zA-z]/g, "eUC");
    },
  },
};
</script>
